package com.rundatop.core.util;

import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics2D;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;
 
import javax.imageio.ImageIO;
 
import org.apache.commons.codec.binary.Base64;
 
import com.google.zxing.BarcodeFormat;
import com.google.zxing.EncodeHintType;
import com.google.zxing.MultiFormatWriter;
import com.google.zxing.WriterException;
import com.google.zxing.common.BitMatrix;
import com.google.zxing.qrcode.decoder.ErrorCorrectionLevel;
 
/**
 * 使用zxing生成带logo的二维码图片，自动调节logo图片相对二维码图片的大小
 * 可选是否带logo，可选是否保存二维码图片；结果返回base64编码的图片数据字符串
 * 页面显示：<img src="data:image/png;base64,${imageBase64QRCode}"/>
 * maven jar
 * <dependency>
	    <groupId>com.google.zxing</groupId>
	    <artifactId>core</artifactId>
	    <version>3.1.0</version>
	</dependency>
 * @author nhhyx
 * @date 20170314
 */
public class ZXingCodeUtils {
	private static final int QRCOLOR = 0xFF000000;   //默认是黑色
    private static final int BGWHITE = 0xFFFFFFFF;   //背景颜色
 
    public static void main(String[] args) throws WriterException  
    {  
        try  
        {  
        	/* 调用试例1，从项目中获取logo图片，生成的二维码图片也保存到项目指定文件夹中，使用myeclipse等时，生成的二维码图片在tomcat/webapps/项目中 */        	
        	// 生成二维码图片并将base64编码的数据更新到数据库，data:image/png;base64,***
			// logo图片
			/*String logoUrl = request.getSession().getServletContext()
					.getRealPath("/WEB-INF/assets/admin/img/temp/logo.png");
		    File logoFile = new File(logoUrl);
        	// 输出二维码路径
        	String outPath = request.getSession().getServletContext()
					.getRealPath("/WEB-INF/assets/qr"); 
        	String base64Img = ZXingCodeUtils.createQRCode(logoFile, "这是二维码里面的内容，可以是http网址", outPath, 
        			400, 400, "这是二维码图片下的文字，随图片一起生成");*/
        	
        	/* 调用试例2 */ 
        	//String filePath = request.getSession().getServletContext().getRealPath("/") + "resources/images/logoImages/llhlogo.png";
        	/* logo图片路径和二维码输出路径
        	String logoUrl = request.getSession().getServletContext()
					.getRealPath("/WEB-INF/assets/admin/img/temp/logo.png");
        	String outPath = request.getSession().getServletContext()
					.getRealPath("/WEB-INF/assets/qr");*/
        	//filePath是二维码logo的路径，但是实际中我们是放在项目的某个路径下面的，所以路径用上面的，把下面的注释就好
        	String logoPath = "C:\\logo.png";
        	File logoFile = new File(logoPath);
        	String outPath = "C:";
        	createQRCode(logoFile, "2asdf5", outPath, 400, 400, "这是二维码下的文字");
        }  
        catch (Exception e)  
        {  
            e.printStackTrace();  
        }  
    }  
 
    /**
     * 生成带logo的二维码图片
     * @param logoPath 二维码图片中间包含的logo图片文件，如果不存在，则生成不带logo图片的二维码
     * @param content 内容或跳转路径
     * @param outPath 二维码输出路径，如果为""则表示不输出图片到指定位置，只返回base64图片字符串
     * @param qrImgWidth 二维码图片宽度
     * @param qrImgHeight 二维码图片高度（有文字的话会加高45px）
     * @param productName 二维码图片下的文字
     * @return
     */
    public static String createQRCode(File logoFile, String content, String outPath, 
    		int qrImgWidth, int qrImgHeight, String productName)
    {
        try
        {  
        	ZXingCodeUtils zp = new ZXingCodeUtils();
        	// 生成二维码bufferedImage图片
            BufferedImage bim = zp.getQRCODEBufferedImage(content, BarcodeFormat.QR_CODE, qrImgWidth, qrImgHeight, zp.getDecodeHintType());
            
            // 如果有文字，则二维码图片高度增加45px
            if(!"".equals(productName)){
            	qrImgHeight += 45;
        	}
            // 给二维码图片添加Logo并保存到指定位置，返回base64编码的图片数据字符串
            return zp.createLogoQRCode(outPath, qrImgWidth, qrImgHeight, bim, logoFile, new LogoConfig(), productName);
        }
        catch (Exception e)
        {
            e.printStackTrace();
        }
        return null;
    }
 
    /**
     * 给二维码图片添加Logo图片并生成最终二维码图片
     * @param outPath 输出二维码图片的路径，如果为""则表示不输出图片到指定位置，只返回base64图片字符串
     * @param qrImgWidth 生成二维码图片的宽度
     * @param qrImgHeight 生成二维码图片的高度
     * @param bim 读取二维码图片BufferedImage对象
     * @param logoPic logo图片File文件
     * @param logoConfig logo配置
     * @param productName 二维码图片下的文字
     * @return 返回图片base64编码后的字符串
     */
    public String createLogoQRCode(String outPath, int qrImgWidth, int qrImgHeight, BufferedImage bim, File logoPic, LogoConfig logoConfig, String productName)
    {
        try
        {
            /**
             * 读取二维码图片，并构建绘图对象
             */
            BufferedImage image = bim;
            
            // 如果logo图片存在，则加入到二维码图片中
            if(logoPic != null && logoPic.exists()){
	            Graphics2D g = image.createGraphics();
	
	            /**
	             * 读取Logo图片
	             */
	            BufferedImage logo = ImageIO.read(logoPic);
	            /**
	             * 设置logo的大小,本人设置为二维码图片的20%,因为过大会盖掉二维码
	             */
	            int widthLogo = logo.getWidth(null)>image.getWidth()*3/10?(image.getWidth()*3/10):logo.getWidth(null), 
	                heightLogo = logo.getHeight(null)>image.getHeight()*3/10?(image.getHeight()*3/10):logo.getWidth(null);
	
	            /**
	             * logo放在中心
	             */
	             int x = (image.getWidth() - widthLogo) / 2;
	             int y = (image.getHeight() - heightLogo) / 2;
	             /**
	             * logo放在右下角
	             *  int x = (image.getWidth() - widthLogo);
	             *  int y = (image.getHeight() - heightLogo);
	             */
 
	            //开始绘制图片
	            g.drawImage(logo, x, y, widthLogo, heightLogo, null);
//            	g.drawRoundRect(x, y, widthLogo, heightLogo, 15, 15);
//            	g.setStroke(new BasicStroke(logoConfig.getBorder()));
//            	g.setColor(logoConfig.getBorderColor());
//            	g.drawRect(x, y, widthLogo, heightLogo);
            	g.dispose();
            	
            	logo.flush();
            }
 
            //把商品名称添加上去，商品名称不要太长，这里最多支持两行。太长就会自动截取
            if (productName != null && !productName.equals("")) {
                //新的图片，把带logo的二维码下面加上文字
                BufferedImage outImage = new BufferedImage(qrImgWidth, qrImgHeight, BufferedImage.TYPE_4BYTE_ABGR);
                Graphics2D outg = outImage.createGraphics();
                //画二维码到新的面板
                outg.drawImage(image, 0, 0, image.getWidth(), image.getHeight(), null);
                //画文字到新的面板
                outg.setColor(Color.BLACK); 
                outg.setFont(new Font("宋体",Font.BOLD, 26)); //字体、字型、字号 
                int strWidth = outg.getFontMetrics().stringWidth(productName);
                if (strWidth > 399) {
//                  //长度过长就截取前面部分
//                  outg.drawString(productName, 0, image.getHeight() + (outImage.getHeight() - image.getHeight())/2 + 5 ); //画文字
                    //长度过长就换行
                    String productName1 = productName.substring(0, productName.length()/2);
                    String productName2 = productName.substring(productName.length()/2, productName.length());
                    int strWidth1 = outg.getFontMetrics().stringWidth(productName1);
                    int strWidth2 = outg.getFontMetrics().stringWidth(productName2);
                    outg.drawString(productName1, 200  - strWidth1/2, image.getHeight() + (outImage.getHeight() - image.getHeight())/2 + 12 );
                    BufferedImage outImage2 = new BufferedImage(400, 485, BufferedImage.TYPE_4BYTE_ABGR);
                    Graphics2D outg2 = outImage2.createGraphics();
                    outg2.drawImage(outImage, 0, 0, outImage.getWidth(), outImage.getHeight(), null);
                    outg2.setColor(Color.BLACK); 
                    outg2.setFont(new Font("宋体", Font.BOLD, 26)); //字体、字型、字号 
                    outg2.drawString(productName2, 200  - strWidth2/2, outImage.getHeight() + (outImage2.getHeight() - outImage.getHeight())/2 + 5 );
                    outg2.dispose(); 
                    outImage2.flush();
                    outImage = outImage2;
                }else {
                    outg.drawString(productName, 200  - strWidth/2 , image.getHeight() + (outImage.getHeight() - image.getHeight())/2 + 12 ); //画文字 
                }
                outg.dispose(); 
                outImage.flush();
                image = outImage;
            }
            //logo.flush();
            image.flush();
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            baos.flush();
            ImageIO.write(image, "png", baos);
 
            // 如果输出路径为空，则不保存二维码图片到指定路径下
            if(!"".equals(outPath.trim())){
	            //二维码生成的路径，但是实际项目中，我们是把这生成的二维码显示到界面上的，因此下面的折行代码可以注释掉
	            //可以看到这个方法最终返回的是这个二维码的imageBase64字符串
	            //前端用 <img src="data:image/png;base64,${imageBase64QRCode}"/> 其中${imageBase64QRCode}对应二维码的imageBase64字符串
	            ImageIO.write(image, "png", new File(outPath + "\\" + new Date().getTime() + ".png"));   
            }
            
            // 获取base64编码的二维码图片字符串
            String imageBase64QRCode = Base64.encodeBase64String(baos.toByteArray());
            baos.close();
            return imageBase64QRCode;
        }
        catch (Exception e)
        {
            e.printStackTrace();
        }
        return null;
    }
 
    /**
     * 构建初始化二维码
     *
     * @param bm
     * @return
     */
    public BufferedImage fileToBufferedImage(BitMatrix bm)
    {
        BufferedImage image = null;
        try
        {
            int w = bm.getWidth(), h = bm.getHeight();
            image = new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB);
 
            for (int x = 0; x < w; x++)
            {
                for (int y = 0; y < h; y++)
                {
                    image.setRGB(x, y, bm.get(x, y) ? 0xFF000000 : 0xFFCCDDEE);
                }
            }
 
        }
        catch (Exception e)
        {
            e.printStackTrace();
        }
        return image;
    }
 
    /**
     * 生成二维码bufferedImage图片
     *
     * @param content
     *            编码内容
     * @param barcodeFormat
     *            编码类型
     * @param width
     *            图片宽度
     * @param height
     *            图片高度
     * @param hints
     *            设置参数
     * @return
     */
    public BufferedImage getQRCODEBufferedImage(String content, BarcodeFormat barcodeFormat, int width, int height, Map<EncodeHintType, ?> hints)
    {
        MultiFormatWriter multiFormatWriter = null;
        BitMatrix bm = null;
        BufferedImage image = null;
        try
        {
            multiFormatWriter = new MultiFormatWriter();
            // 参数顺序分别为：编码内容，编码类型，生成图片宽度，生成图片高度，设置参数
            bm = multiFormatWriter.encode(content, barcodeFormat, width, height, hints);
            int w = bm.getWidth();
            int h = bm.getHeight();
            image = new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB);
 
            // 开始利用二维码数据创建Bitmap图片，分别设为黑（0xFFFFFFFF）白（0xFF000000）两色
            for (int x = 0; x < w; x++)
            {
                for (int y = 0; y < h; y++)
                {
                    image.setRGB(x, y, bm.get(x, y) ? QRCOLOR : BGWHITE);
                }
            }
        }
        catch (WriterException e)
        {
            e.printStackTrace();
        }
        return image;
    }
 
    /**
     * 设置二维码的格式参数
     *
     * @return
     */
    public Map<EncodeHintType, Object> getDecodeHintType()
    {
        // 用于设置QR二维码参数
        Map<EncodeHintType, Object> hints = new HashMap<EncodeHintType, Object>();
        // 设置QR二维码的纠错级别（H为最高级别）具体级别信息
        hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.H);
        // 设置编码方式
        hints.put(EncodeHintType.CHARACTER_SET, "utf-8");
        hints.put(EncodeHintType.MARGIN, 0);
        hints.put(EncodeHintType.MAX_SIZE, 350);
        hints.put(EncodeHintType.MIN_SIZE, 100);
 
        return hints;
    }
}
 
/**
 * logo的配置
 * @author nhhyx
 *
 */
class LogoConfig
{
    // logo默认边框颜色
    public static final Color DEFAULT_BORDERCOLOR = Color.WHITE;
    // logo默认边框宽度
    public static final int DEFAULT_BORDER = 2;
    // logo大小默认为照片的1/5
    public static final int DEFAULT_LOGOPART = 5;
 
    private final int border = DEFAULT_BORDER;
    private final Color borderColor;
    private final int logoPart;
 
    /**
     * Creates a default config with on color {@link #BLACK} and off color
     * {@link #WHITE}, generating normal black-on-white barcodes.
     */
    public LogoConfig()
    {
        this(DEFAULT_BORDERCOLOR, DEFAULT_LOGOPART);
    }
 
    public LogoConfig(Color borderColor, int logoPart)
    {
        this.borderColor = borderColor;
        this.logoPart = logoPart;
    }
 
    public Color getBorderColor()
    {
        return borderColor;
    }
 
    public int getBorder()
    {
        return border;
    }
 
    public int getLogoPart()
    {
        return logoPart;
    }
}
